使用Alamofire发送带有简单字符串的POST请求

100

如何在我的iOS应用程序中使用Alamofire发送带有简单字符串的POST请求?

作为默认设置,Alamofire需要请求参数:

Alamofire.request(.POST, "http://mywebsite.example/post-request", parameters: ["foo": "bar"])

这些参数包含键值对。但我不想在HTTP主体中发送键值字符串。

我的意思是这样的:

Alamofire.request(.POST, "http://mywebsite.example/post-request", body: "myBodyString")
12个回答

93

你的示例 Alamofire.request(.POST, "http://mywebsite.example/post-request", parameters: ["foo": "bar"]) 已经将字符串 "foo=bar" 作为请求体。 但如果你真的想要自定义格式的字符串,可以这样做:

Alamofire.request(.POST, "http://mywebsite.example/post-request", parameters: [:], encoding: .Custom({
            (convertible, params) in
            var mutableRequest = convertible.URLRequest.copy() as NSMutableURLRequest
            mutableRequest.HTTPBody = "myBodyString".dataUsingEncoding(NSUTF8StringEncoding, allowLossyConversion: false)
            return (mutableRequest, nil)
        }))

注意:parameters不应该为nil

更新(Alamofire 4.0,Swift 3.0):

在Alamofire 4.0中,API已经改变。因此对于自定义编码,我们需要一个遵守ParameterEncoding协议的值/对象。

extension String: ParameterEncoding {

    public func encode(_ urlRequest: URLRequestConvertible, with parameters: Parameters?) throws -> URLRequest {
        var request = try urlRequest.asURLRequest()
        request.httpBody = data(using: .utf8, allowLossyConversion: false)
        return request
    }

}

Alamofire.request("http://mywebsite.example/post-request", method: .post, parameters: [:], encoding: "myBody", headers: [:])

这个完美地工作,不仅适用于简单的JSON类型映射字符串,而且适用于所有类型的JSON映射字符串。 - Sahil Kapoor
1
@silmaril 在我的情况下,只有使用PUT方法才能从后端获取原始请求,你能帮我看看为什么POST请求看不到任何东西吗? - A.J.
1
在Alamofire 4 Swift 3中,.Custom已不再可用,我找不到替代方法,有什么提示吗? - Raheel Sadiq
@Silmaril,我在这里发布了一个类似的问题,如果您可以请分享一个答案https://dev59.com/wZzha4cB1Zd3GeqPAygG - mike vorisis
运行完美。应该标记为正确答案。 - Baylor Mitchell
显示剩余10条评论

76

您可以这样做:

  1. 我创建了一个独立的Alamofire请求对象。
  2. 将字符串转换为数据
  3. 在httpBody中放入数据

var request = URLRequest(url: URL(string: url)!)
request.httpMethod = HTTPMethod.post.rawValue
request.setValue("application/json", forHTTPHeaderField: "Content-Type")

let pjson = attendences.toJSONString(prettyPrint: false)
let data = (pjson?.data(using: .utf8))! as Data

request.httpBody = data

Alamofire.request(request).responseJSON { (response) in


    print(response)

}

6
这应该是被采纳的答案。它简单易懂,恰到好处地完成了所需功能,没有不必要的扩展或转换。谢谢,伙计。 - Sea Coast of Tibet
2
我已经“借用”了你在另一个问题中的答案 - https://dev59.com/jlwY5IYBdhLWcg3w_MCW#42411188 - Sea Coast of Tibet
3
出勤记录是什么,能否请您发布一个更完整的代码片段? - SyraKozZ
@SyraKozZ 不管出席情况如何,唯一重要的是pjson是一个JSON字符串。您可以在那里放置任何JSON字符串。 - afrodev

19

如果你使用Alamofire,只需将encoding类型设置为URLEncoding.httpBody即可。

这样,即使在代码中定义为json,你也可以将数据作为字符串发送到httpbody。

对我来说有效。

Bad Filali的问题更新:

var url = "http://..."
let _headers : HTTPHeaders = ["Content-Type":"application/x-www-form-urlencoded"]
let params : Parameters = ["grant_type":"password","username":"mail","password":"pass"]

let url =  NSURL(string:"url" as String)

request(url, method: .post, parameters: params, encoding: URLEncoding.httpBody, headers: _headers).responseJSON(
    completionHandler: { response in response
        let jsonResponse = response.result.value as! NSDictionary
        
        if jsonResponse["access_token"] != nil
        {
            access_token = String(describing: jsonResponse["accesstoken"]!)
        }
    })

1
我会更新我的答案并写出我使用的代码,因为我找不到如何在这里将代码作为评论编写。抱歉回复晚了。@Badr Filali - Cemal BAYRI
1
对我来说起作用了,但我的body需要是一个JSON,所以我将编码从URLEncoding.httpBody改为encoding: JSONEncoding.default,一切都很好运行。 - Ângelo Polotto
谢谢您的回答。但我面临的问题是,当我发送一个变量而不是“双引号字符串”时,会收到错误代码400。我该如何解决这个问题? - viper
你能否发送给我示例代码,也许这样我可以更好地帮助你 @viper - Cemal BAYRI
你有解包 "edtMail.text" 吗?也许你的文本框或其他项目是 nil 了? - Cemal BAYRI
显示剩余4条评论

8

我修改了 @Silmaril 的答案,扩展了 Alamofire 的 Manager。 这个解决方案使用 EVReflection 直接序列化一个对象:

//Extend Alamofire so it can do POSTs with a JSON body from passed object
extension Alamofire.Manager {
    public class func request(
        method: Alamofire.Method,
        _ URLString: URLStringConvertible,
          bodyObject: EVObject)
        -> Request
    {
        return Manager.sharedInstance.request(
            method,
            URLString,
            parameters: [:],
            encoding: .Custom({ (convertible, params) in
                let mutableRequest = convertible.URLRequest.copy() as! NSMutableURLRequest
                mutableRequest.HTTPBody = bodyObject.toJsonString().dataUsingEncoding(NSUTF8StringEncoding, allowLossyConversion: false)
                return (mutableRequest, nil)
            })
        )
    }
}

然后您可以像这样使用它:

Alamofire.Manager.request(.POST, endpointUrlString, bodyObject: myObjectToPost)

8

根据Illya Krit的答案

细节

  • Xcode版本10.2.1(10E1001)
  • Swift 5
  • Alamofire 4.8.2

解决方案

import Alamofire

struct BodyStringEncoding: ParameterEncoding {

    private let body: String

    init(body: String) { self.body = body }

    func encode(_ urlRequest: URLRequestConvertible, with parameters: Parameters?) throws -> URLRequest {
        guard var urlRequest = urlRequest.urlRequest else { throw Errors.emptyURLRequest }
        guard let data = body.data(using: .utf8) else { throw Errors.encodingProblem }
        urlRequest.httpBody = data
        return urlRequest
    }
}

extension BodyStringEncoding {
    enum Errors: Error {
        case emptyURLRequest
        case encodingProblem
    }
}

extension BodyStringEncoding.Errors: LocalizedError {
    var errorDescription: String? {
        switch self {
            case .emptyURLRequest: return "Empty url request"
            case .encodingProblem: return "Encoding problem"
        }
    }
}

使用方法

Alamofire.request(url, method: .post, parameters: nil, encoding: BodyStringEncoding(body: text), headers: headers).responseJSON { response in
     print(response)
}

非常感谢您! - Kerim Khasbulatov

5
如果您想在请求中作为原始主体发布字符串。
return Alamofire.request(.POST, "http://mywebsite.com/post-request" , parameters: [:], encoding: .Custom({
            (convertible, params) in
            let mutableRequest = convertible.URLRequest.copy() as! NSMutableURLRequest

            let data = ("myBodyString" as NSString).dataUsingEncoding(NSUTF8StringEncoding)
            mutableRequest.HTTPBody = data
            return (mutableRequest, nil)
        }))

嗨JITHINRAJ,我在这里发布了一个问题https://dev59.com/wZzha4cB1Zd3GeqPAygG 我想发送一个XML主体。 - mike vorisis

5

我已经为字符串数组完成了这项任务。这个解决方案是为正文中的字符串调整过的。

从 Alamofire 4 开始,这是一种“原生”的方法:

struct JSONStringArrayEncoding: ParameterEncoding {
    private let myString: String

    init(string: String) {
        self.myString = string
    }

    func encode(_ urlRequest: URLRequestConvertible, with parameters: Parameters?) throws -> URLRequest {
        var urlRequest = urlRequest.urlRequest

        let data = myString.data(using: .utf8)!

        if urlRequest?.value(forHTTPHeaderField: "Content-Type") == nil {
            urlRequest?.setValue("application/json", forHTTPHeaderField: "Content-Type")
        }

        urlRequest?.httpBody = data

        return urlRequest!
    }
}

然后使用以下方式发出请求:

Alamofire.request("your url string", method: .post, parameters: [:], encoding: JSONStringArrayEncoding.init(string: "My string for body"), headers: [:])

4

我参考了@afrodev的答案。在我的情况下,我将参数作为字符串传递给我的函数,这个字符串需要在请求中发送。所以,这是代码:

func defineOriginalLanguage(ofText: String) {
    let text =  ofText
    let stringURL = basicURL + "identify?version=2018-05-01"
    let url = URL(string: stringURL)

    var request = URLRequest(url: url!)
    request.httpMethod = HTTPMethod.post.rawValue
    request.setValue("text/plain", forHTTPHeaderField: "Content-Type")
    request.httpBody = text.data(using: .utf8)

    Alamofire.request(request)
        .responseJSON { response in
            print(response)
    }
}

你到底不明白什么? - GeoSD

3
let parameters = ["foo": "bar"]
              
    // All three of these calls are equivalent
    AF.request("https://httpbin.org/post", method: .post, parameters: parameters)
    AF.request("https://httpbin.org/post", method: .post, parameters: parameters, encoder: URLEncodedFormParameterEncoder.default)
    AF.request("https://httpbin.org/post", method: .post, parameters: parameters, encoder: URLEncodedFormParameterEncoder(destination: .httpBody))
    
    

2
func paramsFromJSON(json: String) -> [String : AnyObject]?
{
    let objectData: NSData = (json.dataUsingEncoding(NSUTF8StringEncoding))!
    var jsonDict: [ String : AnyObject]!
    do {
        jsonDict = try NSJSONSerialization.JSONObjectWithData(objectData, options: .MutableContainers) as! [ String : AnyObject]
        return jsonDict
    } catch {
        print("JSON serialization failed:  \(error)")
        return nil
    }
}

let json = Mapper().toJSONString(loginJSON, prettyPrint: false)

Alamofire.request(.POST, url + "/login", parameters: paramsFromJSON(json!), encoding: .JSON)

什么是Mapper? - Ahmadreza

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接